8. Service 层

除了 Web 层 之外, Grails 还定义了service 层的概念。Grails 团队不赞成在controllers中嵌入核心的应用程序逻辑,因为这样并没有提升重用和清楚的关注点分离。

Grails中的Services在应用程序中被视为放置多数逻辑的地方 。 从controllers脱离,负责处理通过重定向的请求流等等。

创建Service

你可以在终端窗口的项目根目录下运行 create-service 创建Service:

grails create-service simple

上面的示例将在grails-app/services/SimpleService.groovy位置创建一个Service。 service的名字按规约以 Service结尾。 除此之外,service就是个普通的Groovy类:

class SimpleService {	
}

8.1 声明式事务处理

Services一般涉及协调 domain 类之间的逻辑, , 因此常常涉及大范围的持久化操作。 因为services性质,它们常常需要事物状态。你可以使用 withTransaction方法来编程事物,不过,这是重复性的,没有充分利用Spring强大的潜在事物抽象

Services允许启用事物,本质上是以声明的方式来声明service中的所有方法必须用于事物。默认情况下,所有services的事物都是可用的——禁用它,只需设置 transactional属性为 false:

class CountryService {
    static transactional = false
}

你也可以默认设置这个属性为 true 在以后改变它,或者清楚的表明这个服务是有意地用于事物。

警告: 依赖注入唯一 声明事物工作的方式。你不能使用new操作符,像这样new BookService()获取事物服务

其结果是,所有的方法都被包含在事物中,当方法体中抛出异常时,自动回滚。事物的传播级别被默认设置为 PROPAGATION_REQUIRED.

8.2 服务作用域

默认情况下,存取服务方法是非同步的,所以无法阻止同步执行这些函数。事实上,因为服务是单例的,可以被同时使用,你必须非常小心服务中存储状态。或者采用容易(和更好的)途径并不在y service中存储状态。

你可以通过把service放置于特定的作用域来改变这样的行为:

假如你的service为flash, flowconversation 作用域,它需要实现 java.io.Serializable 并只用于 Web Flow上下文

为了启用一个作用域,在你的类中添加一个静态scope属性,其值为上面所述的作用域之一:

static scope = "flow"

8.3 依赖注入与服务

依赖注入基础

Grails服务的一个重要方面是,有能力利用Spring 框架的依赖注入能力。 Grails支持 "依赖注入通过规约". 换句话说,你可以使用一个属性名表示的一个服务的类名,自动把他们注入到 controllers, tag libraries,等等。

作为示例,给定的服务名为BookService, 如果你像下面这样在controller中放置一个名为bookService 的属性:

class BookController {
   def bookService
   …
}

在这种情况下,Spring 容器将自动注入一个基于它自己配置作用域的服务实体。 所有的依赖注入是通过名字的; Grails 不支持类型注入。 你也可以像下面这样指定类型:

class AuthorService {
	BookService bookService
}

不过, 存在副作用,即在开发模式下BookService的改变会在加载时抛出一个错误。

依赖注入与服务

你可以使用相同的技术在一个服务中注入另一个服务。 如果说,你的AuthorService需要一个 BookService, 可以像下面这样声明 AuthorService:

class AuthorService {
	def bookService
}

依赖注入与Domain类

你甚至可以在domain类中注入服务, 这可以帮助开发出各种丰富的domain:

class Book {	
	…
	def bookService
	def buyBook() {
		bookService.buyBook(this)
	}
}

8.4 Using Services from Java

服务的强大在于它包含了可重用的逻辑,你可以使用来自其他类的服务,包括Java类。这里有一些方法让你重用来自Java的服务。 简单的方法是把你的服务移动到grails-app/services目录下的一个包里。 这是关键步骤,因为你不可能在Java中导入一个默认package 。作为示例, BookService 就是因为上面的原因,在下面Java中不能使用:

class BookService {
	void buyBook(Book book) {
		// logic
	}
}

不过, 把这个类放入一个package中便可修复,, 把这个类移动到 grails-app/services/bookstore子目录, 然后,修改package 声明:

package bookstore
class BookService {
	void buyBook(Book book) {
		// logic
	}
}

package的替代是,定义个需要服务实现的接口:

package bookstore;
interface BookStore {
	void buyBook(Book book);
}

然后,服务:

class BookService implements bookstore.BookStore {
	void buyBook(Book b) {
		// logic
	}
}

后一种方法更熟悉, 在Java端,只需要接口的引用,而不需要实现类。 无论哪种方式,这个练习的目的是,在编译时,让Java能够静态解决类(或接口)的使用。现在,这样便可在 src/java包内创建一个Java类,并提供了一个setter,在Spring中使用bean的类型和它的名字:

package bookstore;
// note: this is Java class
public class BookConsumer {
	private BookStore store;

public void setBookStore(BookStore storeInstance) { this.store = storeInstance; } … }

这样一来,你可以在 grails-app/conf/spring/resources.xml中把这个Java当做Spring bean来配置 (更多详情查看 Grails and Spring):

<bean id="bookConsumer" class="bookstore.BookConsumer">
	<property name="bookStore" ref="bookService" />
</bean>